#pragma once

#include <iostream>
#include <string>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/wait.h>
#include <sys/time.h>
#include <sys/resource.h>


#include "../comm/log.hpp"
#include "../comm/util.hpp"


namespace ns_runner
{
    using namespace ns_log;
    using namespace ns_util;

    class Runner
    {
    public:
        Runner()
        {

        }
        ~Runner()
        {

        }
    public:
        //提供设置进程占用资源大小的接口
        static void SetProcLimit(int _cpu_limit,int _mem_limit)
        {
            //设置CPU时长
            struct rlimit cpu_rlimit;
            cpu_rlimit.rlim_max = RLIM_INFINITY;
            cpu_rlimit.rlim_cur = _cpu_limit;
            setrlimit(RLIMIT_CPU,&cpu_rlimit);

            //设置内存大小
            struct rlimit mem_rlimit;
            mem_rlimit.rlim_max = RLIM_INFINITY;
            mem_rlimit.rlim_cur = _mem_limit * 1024;
            setrlimit(RLIMIT_AS,&mem_rlimit);
        }

        //指明文件名即可,不需要代理路径,不需要带后缀
        /**********************************
         * 返回值 > 0:程序异常了,退出时受到了信号,返回值就是对应的信号编号
         * 返回值 = 0:正常运行完毕的,结果保存到了对应的临时文件中
         * 返回值 < 0:内部错误
         * 
         * cpu_limit : 该程序运行时,可以使用最大的CPU资源
         * mem_limit : 该程序运行时,可以使用最大的内存大小(KB)
         *********************************/
        static int Run(const std::string & file_name,int cpu_limit,int mem_limit)
        {
            /******************************
            * 程序运行:
            * 1. 代码跑完,结果正确
            * 2. 代码跑完,结果不正确
            * 3. 代码没跑完,异常了
            * Run模块需要考虑代码跑完,结果正确与否吗??? 不考虑
            * 结果正确与否: 是由我们的测试用例来决定的!
            * 我们只考虑: 是否正确运行完毕
            * 
            * 
            * 我们必须知道可执行程序是谁? ./temp/code.exe
            * 一个程序在默认启动的时候
            * 标准输入: 不处理(用户自测)
            * 标准输出: 程序运行完成,输出结果是什么
            * 标准错误: 运行时错误信息
            * 全部写道同名文件中
            ******************************/

            std::string _execute = PathUtil::Exe(file_name);
            std::string _stdin   = PathUtil::Stdin(file_name);
            std::string _stdout  = PathUtil::Stdout(file_name);
            std::string _stderr  = PathUtil::Stderr(file_name);

            umask(0);
            int _stdin_fd = open(_stdin.c_str(),O_CREAT | O_RDONLY,0644);
            int _stdout_fd = open(_stdout.c_str(),O_CREAT | O_WRONLY,0644);
            int _stderr_fd = open(_stderr.c_str(),O_CREAT | O_WRONLY,0644);

            if(_stdin_fd < 0 || _stdout_fd < 0 || _stderr_fd < 0)
            {
                LOG(ERROR) << "运行时打开标准文件失败" << "\n";
                return -1;//代表打开文件失败
            }

            pid_t pid = fork();
            if(pid < 0)
            {
                LOG(ERROR) << "运行时创建子进程失败" << "\n";
                close(_stdin_fd);
                close(_stdout_fd);
                close(_stderr_fd);
                return -2;//代表创建子进程失败
            }
            else if(pid == 0)
            {
                dup2(_stdin_fd,0);
                dup2(_stdout_fd,1);
                dup2(_stderr_fd,2);

                SetProcLimit(cpu_limit,mem_limit);

                execl(_execute.c_str()/*我要执行谁*/,_execute.c_str()/*在命令行上怎么执行*/,nullptr);
                exit(1);
            }
            else
            {
                close(_stdin_fd);
                close(_stdout_fd);
                close(_stderr_fd);
                int status = 0;
                waitpid(pid,&status,0);
                LOG(INFO) << "运行完毕, info: " << (status & 0x7F) << "\n";
                //程序运行异常,一定是因为受到了信号
                return status & 0x7F; //将程序是否受到异常(信号 1-30)进行返回
            }
        }
    };
}





